home *** CD-ROM | disk | FTP | other *** search
/ TeX 1995 July / TeX CD-ROM July 1995 (Disc 1)(Walnut Creek)(1995).ISO / biblio / bibtex / contrib / phy-bstyles / cpp.el < prev    next >
Lisp/Scheme  |  1992-06-14  |  7KB  |  200 lines

  1. ;; A C preprocessor written for GnuEmacs for use in generating BibTeX style
  2. ;; files from a master file.  In addition it removes whole-line comments (with
  3. ;; "%" in the first column) and collapses multiple blank lines to a single
  4. ;; blank line.
  5.  
  6. ;;  Charles Karney
  7. ;;  Plasma Physics Laboratory    Phone:     +1 609 243 2607
  8. ;;  Princeton University    E-mail:     Karney@Princeton.EDU
  9. ;;  PO Box 451
  10. ;;  Princeton, NJ 08543-0451
  11.  
  12. ;; RESTRICTIONS:
  13.  
  14. ;; Recognizes only a subset of C precessor symbols:
  15. ;;     #include, #ifdef, #ifndef, #define, #undef, #if, #else, #endif.
  16.  
  17. ;; Macro values can only be numbers.
  18.  
  19. ;; Expressions are not allowed, except for !MACRO.  Thus
  20. ;;     "#if FOO" and "#if !FOO" work OK
  21. ;;     "#if (FOO | BAR)" and "#if FOO == 4" do not work
  22.  
  23. ;; Value substitution is done in a separate pass through the file at the end.
  24. ;; This means that the value substituted for a particular macro is the last
  25. ;; one defined.  Thus
  26. ;;     #define FOO 4
  27. ;;     FOO
  28. ;;     #define FOO 10
  29. ;;     FOO
  30. ;; produces
  31. ;;     10
  32. ;;     10
  33.  
  34. (defvar cpp-macros nil "currently defined macros")
  35. (defvar cpp-values nil "values for currently defined macros")
  36.  
  37. (defun cpp (init)
  38.   "Run C preprocessor on current buffer.  Argument is single macro to
  39. get defined before processing begins.
  40. Recognizes a subset of C precessor symbols:
  41.   #include, #ifdef, #ifndef, #define, #if, #else, #endif.
  42. Also strips out any comments starting with % in the first column."
  43.   (interactive "sRun cpp defining: ")
  44.   (goto-char (point-min))
  45.   (setq init (upcase init))
  46.   (let (verb)
  47.       (setq cpp-macros nil cpp-values nil)
  48.       (cond ((> (length init) 0)
  49.          (cpp-define init 1)
  50.          (insert (concat "%% #define " init
  51.                  " 1\t\t% " (current-time-string) "\n"))))
  52.       (insert (concat "%% #include \"" (buffer-file-name) "\"\n"))
  53.       (while (re-search-forward "^#" nil t)
  54.     (save-excursion (beginning-of-line) (insert "%% "))
  55.     (setq verb (cpp-next-word))
  56.     (cond ((equal verb "include") (cpp-include (cpp-next-word)))
  57.           ((equal verb "define")
  58.            (cpp-define (cpp-next-word) (cpp-eval (cpp-next-word))))
  59.           ((equal verb "undef")
  60.            (cpp-undef (cpp-next-word)))
  61.           ((equal verb "ifdef")
  62.            (cond ((zerop (cpp-ifdef (cpp-next-word))) (cpp-skip))))
  63.           ((equal verb "ifndef")
  64.            (cond ((not (zerop (cpp-ifdef (cpp-next-word)))) (cpp-skip))))
  65.           ((equal verb "if")
  66.            (cond ((zerop (cpp-eval (cpp-next-word))) (cpp-skip))))
  67.           ((equal verb "else")
  68.            (cpp-skip))
  69.           ((equal verb "endif"))
  70.           (t (error "Unknown preprocessor directive: %s" verb)))))
  71.     (goto-char (point-min))
  72.     (forward-line 2)
  73.     (while (re-search-forward "^%" nil t)
  74.       (beginning-of-line)
  75.       (delete-region (point) (progn (forward-line 1) (point))))
  76.     (goto-char (point-min))
  77.     (while (re-search-forward "\n\n\n" nil t)
  78.       (backward-char 3)
  79.       (delete-char 1))
  80.     (let ((macros cpp-macros) (values cpp-values) (case-fold-search nil))
  81.       (while macros
  82.     (goto-char (point-min))
  83.     (forward-line 2)
  84.     (while (re-search-forward (concat "\\b" (car macros) "\\b") nil t)
  85.       (delete-region (match-beginning 0) (match-end 0))
  86.       (insert (int-to-string (car values))))
  87.     (setq macros (cdr macros) values (cdr values)))))
  88.  
  89. (defun cpp-include (file)
  90.   "Include a file"
  91.   (forward-line 1)
  92.   (insert-file (substring file 1 -1)))
  93.  
  94. (defun cpp-eval (macro)
  95.   "Returns the value of a macro"
  96.   (cond ((equal (substring macro 0 1) "!")
  97.      (if (eq (cpp-eval (substring macro 1 nil)) 0)
  98.          1
  99.        0))
  100.     ((or (equal (substring macro 0 1) "0")
  101.          (not (eq (string-to-int macro) 0)))
  102.      (string-to-int macro))
  103.     (t (let ((macros cpp-macros) (values cpp-values))
  104.          (while (not (or (null macros)
  105.                  (equal (car macros) macro)))
  106.            (setq macros (cdr macros)
  107.              values (cdr values)))
  108.          (if (null macros)
  109.          (error "Undefined macro %s",macro)
  110.            (car values))))))
  111.     
  112. (defun cpp-define (macro value)
  113.   "Make a definition"
  114.   (let ((macros cpp-macros) (values cpp-values))
  115.     (while (not (or (null macros)
  116.             (equal (car macros) macro)))
  117.       (setq macros (cdr macros)
  118.         values (cdr values)))
  119.     (if (null macros) (setq cpp-macros (cons macro cpp-macros)
  120.                 cpp-values (cons value cpp-values))
  121.       (rplaca values value))))
  122.  
  123. (defun cpp-undef (macro)
  124.   "Remove a definition"
  125.   (let ((macros cpp-macros) (values cpp-values) macrosa valuesa)
  126.     (cond ((null macros))
  127.       ((equal (car macros) macro)
  128.        (setq cpp-macros (cdr macros)
  129.          cpp-values (cdr values)))
  130.       (t (setq macrosa (cdr macros) valuesa (cdr values))
  131.          (while (not (or (null macrosa)
  132.                  (equal (car macrosa) macro)))
  133.            (setq macros macrosa
  134.              values valuesa
  135.              macrosa (cdr macrosa)
  136.              valuesa (cdr valuesa)))
  137.          (cond ((null macros))
  138.            (t (rplacd macros (cdr macrosa))
  139.               (rplacd values (cdr valuesa))))))))
  140.  
  141. (defun cpp-ifdef (macro)
  142.   "Returns 1 if macro is defined, 0 otherwise"
  143.   (let ((macros cpp-macros) (values cpp-values))
  144.     (while (not (or (null macros)
  145.             (equal (car macros) macro)))
  146.       (setq macros (cdr macros)
  147.         values (cdr values)))
  148.     (if (null macros) 0 1)))
  149.  
  150. (defun cpp-next-word ()
  151.   "Return next blank-delimited word in buffer"
  152.   (skip-chars-forward " \t")
  153.   (buffer-substring (point)
  154.             (progn (re-search-forward " \\|\t\\|$")
  155.                (match-beginning 0))))
  156.  
  157. (defun cpp-skip ()
  158.   "Skips to next endif or else"
  159.   (forward-line 1)
  160.   (delete-region
  161.    (point)
  162.    (let ((count 1))
  163.      (while (> count 0)
  164.        (re-search-forward "^#[ \t]*\\(if\\|else\\|endif\\)")
  165.        (goto-char (match-beginning 1))
  166.        (cond ((looking-at "if") (setq count (1+ count)))
  167.          ((looking-at "else") (if (eq count 1) (setq count 0)))
  168.          ((looking-at "endif") (setq count (1- count)))))
  169.      (beginning-of-line)
  170.      (point)))
  171.   (insert "%% "))
  172.  
  173. (defun cpp-file (name)
  174.   "Run C preprocessor on physics.btx.  Argument is single macro to
  175. get defined before processing begins and this is used in the filename
  176. that the results get written to."
  177.   (interactive "sRun cpp-file defining: ")
  178.   (let ((indir "tex$root:[bibtex]")
  179.     (outdir "tex$root:[latex]"))
  180.     (find-file (concat indir "physics.btx"))
  181.     (cpp name)
  182.     (write-file (concat outdir name ".bst"))))
  183.  
  184. (defun cpp-everything ()
  185.   "Runs cpp on physics.btx to produce all standard styles"
  186.   (interactive)
  187.   (let ((macros (append
  188.          (list "aip" "pf" "nf" "nflet" "iaea" "cpc" "rmp"
  189.                "report" "ppcf" "jcp")
  190. ;         (list "apalike")
  191. ;         (list "plain" "unsrt" "alpha" "abbrv")
  192.          )))
  193.                     ; The standard styles are
  194.                     ; "plain" "unsrt" "abbrv" "alpha"
  195.                     ; A semi-standard style is "apalike"
  196.     (while macros
  197.       (cpp-file (car macros))
  198.       (kill-buffer (buffer-name))
  199.       (setq macros (cdr macros)))))
  200.